/*
 *  linux/fs/ioctl.c
 *
 *  (C) 1991  Linus Torvalds
 */

#include <string.h>     // 字符串头文件 主要定义了一些有关字符串操作的嵌入函数
#include <errno.h>      // 错误号头文件 包含系统中各种出错号（Linus从MINIX中引进的）
#include <sys/stat.h>   // 文件状态头文件 含有文件或文件系统状态结构stat和常量

#include <linux/sched.h>    // 调度程序头文件 定义任务结构task_struct 初始任务0的数据
// 终端 ioctl
extern int tty_ioctl(int dev, int cmd, int arg);
// 定义输入输出控制（ioctl）函数指针类型
typedef int (*ioctl_ptr)(int dev,int cmd,int arg);
// 定义系统中设置种数
#define NRDEVS ((sizeof (ioctl_table))/(sizeof (ioctl_ptr)))
// ioctl 操作函数指针表
static ioctl_ptr ioctl_table[]={
	NULL,		/* nodev */
	NULL,		/* /dev/mem */
	NULL,		/* /dev/fd */
	NULL,		/* /dev/hd */
	tty_ioctl,	/* /dev/ttyx */
	tty_ioctl,	/* /dev/tty */
	NULL,		/* /dev/lp */
	NULL};		/* named pipes */
	
// 系统调用函数-输入输出控制函数 返回：成功则返回0 否则返回出错码 参数：fd-文件描述符 cmd-命令码 arg-参数
int sys_ioctl(unsigned int fd, unsigned int cmd, unsigned long arg)
{	
	struct file * filp;
	int dev,mode;
    // 如果文件描述符超出可打开的文件数 或者对应描述符的文件结构指针为空 则返回出错码 退出
	if (fd >= NR_OPEN || !(filp = current->filp[fd]))
		return -EBADF;
	mode=filp->f_inode->i_mode;     // 取对应文件的属性 如果该文件不是字符文件 也不是块设备文件 则返回出错码 退出
	if (!S_ISCHR(mode) && !S_ISBLK(mode))
		return -EINVAL;
	dev = filp->f_inode->i_zone[0]; // 从字符或块设备文件的i节点中取设备号 如果设备号大于系统现有的设备数 则返回出错码
	if (MAJOR(dev) >= NRDEVS)
		return -ENODEV;
	if (!ioctl_table[MAJOR(dev)])   // 如果该设备在ioctl函数指针表中没有对应函数 则返回出错码
		return -ENOTTY; // 否则返回实际ioctl函数返回码 成功则返回0 否则返回出错码
	return ioctl_table[MAJOR(dev)](dev,cmd,arg);
}
